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(57) Abstract 

In a method of IP routing lookup in a routing table, comprising entries of arbitrary length prefixes with associated next-hop information 
in a next-hop table, to determine where IP datagrams are to be forwarded, a representation of the routing table is stored, in die form of a 
complete prefix tree (7), denned by the prefixes of all routing table entries. Further, a representation of a bit vector (8) comprising data of a 
cut through the prefix tree (7) at a current depth (D), and an array of pointers, comprising indices to the next-hop table and to a next-level 
chunk, are stored. Said bit-vector (8) is divided into bit-masks and a representation of the bit-masks is stored in a maptable. Then, an 
array of code words, each encoding a row index into the maptable and a pointer offset, and an array of base addresses are stored. Finally, 
the lookup is performed. 
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TITLE: METHOD AND SYSTEM FOR FAST ROUTING LOOKUPS 

5 

Field of the Invention 

The present invention relates generally to method and 
system of IP routing lookups in a routing table, comprising 
entries of arbitrary length prefixes with associated next- 
10 hop information in a next-hop table, to determine where IP 
datagrams are to be forwarded. 

Description of the Prior Art 

Internet is an interconnected set of networks, where - 
15 in each of the constituent networks retains its identity, 
and special mechanisms are needed for communication across 
multiple networks . The constituent networks are referred to 
as subnetworks . 

Each subnetwork in the Internet supports communica- 
20 tion among the devices connected to that subnetwork. 

Further, subnetworks are connected by devices referred to 
as interworking units (IWUs) . 

A particular IWU is a router, which is used to 
connect two networks that may or may not be similar. The 
25 router employs an internet protocol (IP) present in each 
router and each host of the network. 

IP provides a connectionless or datagram service 
between stations. 

Routing is generally accomplished by maintaining a 
30 routing table in each station and router that indicates, 
for each possible destination network, the next router to 
which the IP datagram should be sent. 

IP routers do a routing lookup in a routing table to 
determine where IP datagrams are to be forwarded. The 
35 result of the operation is the next-hop on the path towards 
the destination. An entry in a routing table is conceptual- 
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ly an arbitrary length prefix with associated next-hop 
information. Routing lookups must find the routing entry 
with the longest matching prefix. 

Since IP routing lookups have been inherently slow 
5 and complex, operations with prior art solutions have lead 
to a proliferation of techniques to avoid using them. 
Various link layer switching technologies below IP, IP 
layer bypass methods {disclosed by the Proceedings of the 
Conference on Computer Communications (IEEE Infocom) , San 

10 Francisco, California, March 1996; Proceedings of Gigabit 
Network Workshop, Boston, April 1995; and Proceedings of 
ACM SIGCOMM '95, p. 49-58, Cambridge, Massachusetts, August 
1995) , and the development of alternative network layers 
based on virtual circuit technologies such as ATM, are to 

15 some degree results of a wish to avoid IP routing lookups. 
The use of switching link layers and flow or tag 
switching architectures below the IP level adds complexity 
and redundancy to the network. 

Most current IP router designs use caching techniques 

20 wherein the routing entries of the most recently used des- 
tination addresses are kept in a cache. The technique 
relies on there being enough locality in the traffic so 
that the cache hit rate is sufficiently high and the cost 
of a routing lookup is amortized over several packets. 

25 These caching methods have worked well in the past. 
However, as the current, rapid growth of the Internet 
increases the required size of address caches, hardware 
caches may become uneconomical . 

Traditional implementations of routing tables use a 

30 version of Patricia trees (disclosed by the Journal of the 
ACM, 15(4) : 514-534, October 1968), a data structure 
invented almost thirty years ago, with modifications for 
longest prefix matching. 

A straightforward implementation of Patricia trees 

35 ■ for routing lookup purposes, for example in the NetBSD 1.2 
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implementation, uses 24 bytes for leaves and internal 
nodes. With 40,000 entries, the tree structure alone is 
almost 2 megabytes, and in a perfectly balanced tree 15 or 
16 nodes must be traversed to find a routing entry. 



rule, additional nodes need to be traversed to find the 
proper routing information as it is not guaranteed that the 
initial search will find the proper leaf. There are optimi- 
zations that can reduce the size of a Patricia tree and 

10 improve lookup speeds. Nevertheless, the data structure is 
large and too many expensive memory references are needed 
to search it. Current Internet routing tables are too large 
to fit into on-chip caches and off-chip memory references 
to DRAMs are too slow to support required routing speeds. 

15 An early work on improving IP routing performance by 

avoiding full routing lookups (disclosed by the Proceedings 
of the Conference on Computer Communication (IEEE Infocom) , 
New Orleans, Louisiana, March 1988) found that a small des- 
tination address cache can improve routing lookup perfor- 

20 mance by at least 65 per cent. Less than 10 slots were 

needed to get a hit rate over 90 per cent. Such small des- 
tination address caches are not sufficient with the large 
traffic intensities and number of hosts in today's Inter- 
net . 

25 ATM (asynchronous transfer mode) avoids doing routing 

lookups by having a signalling protocol that passes 
addresses to the network during connection setup. 
Forwarding state, accessed by a virtual circuit identifier 
(VCI), is installed in switches along the path of the 

30 connection during setup. ATM cells contain the VCI which 
can then be used as a direct index into a table with 
forwarding state or as the key to a hash function. The 
routing decision is simpler for ATM . However, when packet 
sizes are larger than 48 bytes, more ATM routing decisions 

35 need to be made. 



5 



In some cases, due to the longest matching prefix 
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Tag switching and flow switching (disclosed by the 
Proceedings of the Conference on Computer Communications 
(IEEE Infocom) f San Francisco, California, March 1996) are 
two IP bypass methods that are meant to be operated over 

5 ATM. The general idea is to let IP control link-level ATM 
hardware that performs actual data forwarding. Special 
purpose protocols (disclosed by Request for Comments RFC 
1953, Internet Engineering Task Force, May 1996) are needed 
between routers to agree on what ATM virtual circuit 

10 identifiers to use and which packet should use which VCI. 

Another approach with the same goal of avoiding IP 
processing is taken in the IP/ ATM architecture (disclosed 
by the Proceedings of Gigabit Network Workshop, Boston, 
April 1995; and the Proceedings of ACM SIGCOMM f 95, p. 49- 

15 58, Cambridge, Massachusetts, August 1995), where an ATM 
backplane connects a number of line cards and routing 
cards. IP processing elements located in the routing cards 
process IP headers. When a packet stream arrives, only the 
first IP header is examined and the later packets are 

20 routed the same way as the first one. The main purpose of 
these shortcuts seems to be to amortize the cost of IP 
processing over many packets. 

IP router designs can use special-purpose hardware to 
do IP processing, as in the IBM router (disclosed by the 

25 ■ Journal of High Speed Networks, 1 ( 4 ): 281-288, 1993). This 
can be an inflexible solution. Any changes in the IP format 
or protocol could invalidate such designs. The flexibility 
of software and the rapid performance increase of general 
purpose processors makes such solutions preferable. 

30 Another hardware approach is to use CAMs to do routing 

lookups (disclosed by the Proceedings of the Conference on 
Computer Communications (IEEE Infocom), volume 3, p. 1382- 
1391, San Francisco, 1993) . This is a fast but expensive 
solution. 
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BBN is currently building a pair of multi-gigabit 
routers that use general purpose processors as forwarding 
engines. Little information has been published so far. The 
idea, however, seems to be to use Alpha processors as 

5 forwarding engines and to do all IP processing in software. 
The Publication Gigabit networking, Addison-Wesley, 
Reading, Massachusetts, 1993, shows that it is possible to 
do IP processing in no more than 200 instructions, assuming 
a hit in a route cache. The secondary cache of the Alpha is 

10 used as a large LRU {least recently used) cache of 

destination addresses. The scheme presumes locality in 
traffic patterns. With low locality the cache hit rate 
could become too low and performance would suffer. 

15 Summary of the Invention 

It is therefore an objective of the present invention 
to provide an improved IP routing lookup method and system 
for performing full routing lookups for each IP packet up 
to gigabit speeds, which method and system overcome the 
20 above mentioned disadvantages. 

A further objective is to increase routing lookup 
speeds with a conventional microprocessor. 

Another objective is to minimize lookup time in the 
forwarding table. 
25 Another further objective of the invention is to 

provide a data structure which can fit entirely in the 
cache of a conventional microprocessor. 

Consequently, memory accesses will be orders 
magnitude faster than if the data structure needs to reside 
30 in memory consisting of, for example, relatively slow DRAM. 
These objectives are obtained with the IP routing 
lookup method and system according to the invention, which 
system is a data structure that can represent large routing 
tables in a very compact form and which can be searched 
35 quickly using few memory references . 



WO 99/14906 PCT/SE98/00854 



Brief Description of the Drawings 

In order to explain the invention in more detail and 
the advantages and features of the invention preferred 
5 embodiments will be described in detail below, reference 
being made to the accompanying drawings, in which 

FIG 1 is a schematical view of a router design, 

FIG 2 is an illustrative view of a binary tree 
spanning the entire IP address space, 
10 FIG 3 is an illustrative view of routing entries 

defining ranges of IP addresses, 

FIG 4 is an illustrative view of a step of expanding 
the prefix tree to be full, 

FIG 5 is an illustrative view of three levels of the 
15 data structure according to the invention, 

FIG 6 is an illustrative view of a part of a cut 
through the prefix tree at depth 16, 

FIG 7 is an illustrative view of a first level search 
of the data structure, 
20 FIG 8 is a table illustrating data on forwarding 

tables constructed from various routing tables, 

FIG 9 is a table illustrating processor and cache 

data, 

FIG 10 is a diagram showing lookup time distribution 
25 for an Alpha 21164, and 

FIG 11 is a diagram showing lookup time distribution 
for a Pentium Pro. 

Detailed Description of the Invention 

30 With reference to FIG 1 a router design comprises a 

number of network inbound interfaces 1, network outbound 
interfaces 2, forwarding engines 3, and a network processor 

4, all of which are interconnected by a connection fabric 

5. Inbound interfaces 1 send packet headers to the 

35 forwarding engines 3 through the connection fabric 5, The 
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forwarding engines 3 in turn determine which outgoing 
interface 2 the packet should be sent to. This information 
is sent back to the inbound interface l f which forwards the 
packet to the outbound interface 2. The only task of a 
5 forwarding engine 3 is to process packet headers. All other 
tasks such as participating in routing protocols, resource 
reservation, handling packets that need extra attention, 
and other administrative duties, are handled by the network 
processor 4. 

10 Each forwarding engine 3 uses a local version of the 

routing table, a forwarding table, downloaded from the 
network processor 4 and stored in storage means in the 
forwarding engine 3, to make its routing decisions. It is 
not necessary to download a new forwarding table for each 

15 routing update. Routing updates can be frequent but since 
routing protocols need some time to converge, forwarding 
tables can grow a little stale and need not change more 
than at most once per second (disclosed by Stanford 
University Workshop on Fast Routing and Switching, December 

20 1996; http://tiny-tera.stanford.edu/Workshop_Dec96/) . 

The network processor 4 needs a dynamic routing table 
designed for fast updates and fast generation of forwarding 
tables. The forwarding tables, on the other hand, can be 
optimized for lookup speed and need not be dynamic. 

25 To minimize lookup time, two parameters, the number 

of memory accesses required during lookup, and the size of 
the data structure/- have to be simultaneously minimized in 
the data structure in the forwarding table. 



30 during a lookup is important because memory accesses are 
relatively slow and usually the bottleneck of lookup 
procedures. If the data structure can be made small enough, 
it can fit entirely in the cache of a conventional micro- 
processor. This means that memory accesses will be orders 

35 of magnitude faster than if the data structure needs to 



Reducing the number of memory accesses required 
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reside in a memory consisting of a relatively slow DRAM, as 
is the case for Patricia trees. 

If the forwarding table does not fit entirely in the 
cache, it is still beneficial if a large fraction of the 

5 table can reside in the cache. Locality in traffic patterns 
will keep the most frequently used pieces of the data 
structure in the cache, so that most lookups will be fast. 
Moreover, it becomes feasible to use fast SRAM for the 
small amount of needed external memory. SRAM is expensive, 

10 and more expensive the faster it is. For a given cost, the 
SRAM can be faster if less is needed. 

As secondary design goals, the data structure should 
need few instructions during lookup and keep the entities 
naturally aligned as much as possible to avoid expensive 

15 instructions and cumbersome bit-extraction operations. 

To determine quantitative design parameters for the 
data structure, a number of large routing tables have been 
investigated, as described later. In these cables there are 
fairly few distinct 40,000 routing entries. If next-hops 

20 are identical, the rest of the routing information is also 
the same, and thus all routing entries specifying the same 
next-hop can share routing information. The number of 
distinct next-hops in the routing table of a router is 
limited by the number of other routers or hosts that can be 

25 reached in one hop, so it is not surprising that these 
numbers can be small even for large backbone routers. 
However, if a router is connected to, for instance, a large 
ATM network, the number of next-hops can be much higher. 

In this embodiment, the forwarding table data struc- 

30 ture is designed to accommodate 2 14 or 16K different next- 
hops, which would be sufficient for most cases. If there 
are fewer than 256 distinct next-hops, so that an index 
into the next-hop table can be stored in a single byte, the 
forwarding tables described here can be modified to occupy 

35 considerably less space in another embodiment. 
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The forwarding table is essentially a tree with three 
levels. Searching one level requires one to four memory 
accesses. Consequently, the maximum number of memory 
accesses is twelve. However, with conventional routing 

5 tables, the vast majority of lookups requires searching one 
or two levels only, so the most likely number of memory 
accesses is eight or less. 

For the purpose of understanding the data structure, 
imagine a binary tree 6 that spans the entire IP address 

10 space, which is illustrated in FIG 2. Its depth is 32, and 
the number of leaves is 2 32 , one for each possible IP 
address. The prefix of a routing table entry defines a path 
in the tree ending in some node. All IP addresses (leaves) 
in the subtree rooted at that node should be routed accor- 

15 ding to that routing entry. In this manner each routing 

table entry defines a range of IP addresses with identical 
routing information. 

If several routing entries cover the same IP address, 
the rule of the longest match is applied; it states that 

20 for a given IP address, the routing entry with the longest 
matching prefix should be used. This situation is illustra- 
ted in FIG 3; the routing entry el is hidden by e2 for 
addresses in the range r. 

The forwarding table is a representation of the 

25 binary tree 6 spanned by all routing entries, a prefix tree 
7. It is required that the prefix tree is full, i.e., that 
each node in the tree has either two or no children. Nodes 
with a single child must be expanded to have two children; 
the children added in this way are always leaves, and their 

30 next-hop information will be the same as the next-hop of 
the closest ancestor with next-hop information, or the 
"undefined" next-hop if no such ancestor exists. 

This procedure, illustrated in FIG 4, increases the 
number of nodes in the prefix tree 7, but allows building a 

35 small forwarding table. However, it is not necessary to 
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actually to build the prefix tree to build the forwarding 
table. The prefix tree is used to simplify the description. 
The forwarding table can be built during a single pass over 
ail routing entries. 

5 A set of routing entries divides the IP address space 

into sets of IP addresses. The problem of finding the 
proper routing information is similar to the interval set 
membership problem (disclosed by the SIAM Journal on 
Computing, 17 ( 1) : 1093-1102, December 1988). In this case, 

10 each interval is defined by a node in the prefix tree and, 
therefore, has properties that can be used to compact the 
forwarding table. For instance, each range of IP addresses 
has a length that is a power of two. 

As shown in FIG 5, level one of the data structure 

15 covers the prefix tree down to depth 16, level two covers 
depths 17 to 24, and level three depths 25 to 32. Wherever 
a part of the prefix tree extends below level 16, a level 
two chunk describes that part of the tree. Similarly, 
chunks at level three describe parts of the prefix tree 

20 that are deeper than 24. The result of searching a level of 
the data structure is either an index into the next-hop 
table or an index into an array of chunks for the next 
level . 

At level 1 of the data structure in FIG 5, for 
25 example, there is a cut through the prefix tree 7 at depth 
16. The cut is stored in a bit-vector, with one bit per 
possible node at depth 16. Thus, 2 16 bits = 64Kbits = 8 
Kbytes are required. To find the bit corresponding to the 
initial part of an IP address, the upper 16 bits of the 
30 address is used as an index into the bit-vector. 

When there is a node in the prefix tree at depth 16, 
the corresponding bit in the vector is set. Also, when the 
tree has a leaf at a depth less than 16, the lowest bit in 
the interval covered by that leaf is set. All other bits 
35 are zero. A bit in the bit vector can thus be 
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a one representing that the prefix tree continues 
below the cut; a root head {bits 6, 12 and 13 in FIG 6), or 

a one representing a leaf at depth 16 or less; a 
genuine head (bits 0, 4, 7, 8, 14 and 15 in FIG 6), or 

5 zero, which means that this value is a member of a 

range covered by a leaf at a depth less than 16 (bits 
1,2,3,5,9,10 and- 11 in FIG 6). Members have the same next- 
hop as the largest head smaller than the member. 

For genuine heads an index to the next-hop table have 

10 to be stored. Members will use the same index as the 

largest head smaller than the member. For root heads an 
index to the level two chunk, that represents the 
corresponding subtree, has to be stored. The head infor- 
mation is encoded in 16-bit pointers stored in an array. 

15 Two bits of the pointer encodes what kind of pointer it is, 
and the 14 remaining bits are either indices into the next- 
hop table or into an array containing level two chunks. 

To find the proper pointer, the ' bit-vector is divided 
into bit-masks of length 16 and there are 2 12 = 4096 of 

20 those. Further, the- position of a pointer in the array is 
obtained by adding three entities: a base index, a 6-bit 
offset and a 4-bit offset. Base index plus 6-bit offset 
determines where the pointers corresponding to a particular 
bit-mask are stored. The 4-bit offset specifies which 

25 pointer among those to retrieve. FIG 7 shows how these 

entities are found. The following paragraphs elaborate on 
the procedure. 

Since bit-masks are generated from a full prefix 
tree, not all combinations of the 16 bits are possible. A 

30 non-zero bit-mask of length 2n can be any combination of 

two bit-masks of length n or the bit-mask with value 1. Let 
a(n) be the number of possible non-zero bit-masks of length 
2 n . a(n) is defined by the recurrence 
a(0) = 1, a(n) = 1 + a{n-l) 2 
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The number of possible bit-masks with length 16 are 
thus a (4) + 1 = 678, the additional one is because the bit- 
mask can be zero. An index into a table with an entry for 
each bit-mask thus only needs 10 bits. 

5 Such a table, maptable, is kept to map bit numbers 

within a bit-mask to 4^-bit offsets. The offset specifies 
how many pointers to skip over to find the wanted one, so 
it is equal to the number of set bits smaller than the bit 
index. These offsets are the same for all forwarding 

10 tables, regardless of what values the pointers happen to 
have. The maptable is constant, it is generated once and 
for all. 

The possible bitmasks have the property that any 
interval of bits whose length is an even power of two, and 
15 starts at a bit index that is a multiple of the same power 
of two, either 1) contain all zeroes or 2) have the lowest 
bit set. 

The mapping from bitmask and bit to offset provided 
by the maptable, together with the prefix expansion that 
20 makes the prefix tree complete and enables using the 

mapping, is the key to achieve the small forwarding table 
size . 

The actual bit-masks are not needed, and instead of 
the bit-vector we keep an array of 16-bit code words 

25 consisting of a 10-bit index into the maptable plus a 6-bit 
offset. A 6-bit offset covers up to 64 pointers, so one 
base index per four code words is needed. There can be at 
most 64K pointers so the base indices need to be at most 16 
bits (2 16 = 64K) . 

30 The following steps of pseudo code, in a procedure 

illustrated in FIG 7, are required to search the first 
level of the data structure, wherein the array of code 
words is called code, and the array of base addresses is 
called base, ix is a first index part of the IP address in 

35 the array of code words, bit is a column index part of the 
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IP address into the maptable, ten is a row index part of a 
codeword into the maptable, bix is a second index part of 
the IP address into the array of base addresses, and pix is 
a pointer into the array of pointers, comprising indices to 
5 the next-hop table as well as indices to the level two 
chunk. 

ix := high 12 bits of IP address 

bit := low 4 of high 16 bits of IP address codeword 
10 := code[ix] 

ten := ten bits f rom , codeword six : = six bits from 
codeword 

bix := high 10 bits of IP address 

pix := base[bix] + six + maptable [ten] [bit ] pointer 
15 levell_pointers [pix] 

Thus, only a few bit extraction, array references, 
and additions are needed. No multiplication or division 
instructions are required except for the implicit multipli- 
20 cations when indexing an array. 

Totally 7 bytes need to be accessed to search the 
first level: a two byte code word, a two byte base address, 
one byte (4 bits, really) in the maptable, and finally a 
two byte pointer. The size of the first level is 8K bytes 
25 for the code word array, 2K bytes for the array of base 

indices, plus a number of pointers. The 5 . 3K bytes required 
by. the maptable are shared among all three levels. 

When the bit-mask is zero or has a single bit set, 
the pointer must be an index into the next-hop table. Such 
30 pointers can be encoded directly into the code word and 
thus the maptable needs not contain entries for bit-masks 
one and zero. The number of maptable entries is thus 
reduced to 676 (indices 0 through 675) . When the ten bits 
in the code word {ten above) are larger than 675, the code 
35 word represents a direct index into the next-hop table. The 
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six bits from the code word are used as the lowest 6 bits 
in the index, and (ten-676) are the upper bits of the 
index. This encoding allows at most (1024-676) x2 6 = 22272 
next-hop indices, which is more than the 16K we are 
designing for. This optimization eliminates three memory 
references when a routing entry is located at depth 12 or 
higher, and reduces the number of pointers in the pointer 
array considerably. The cost is a comparison and a 
conditional branch. 

The mapping is implied by the following data and 
functions in the C programming language. 

The important feature of this code is the mapping it 
provides from an individual bitmask to an array of offsets. 

Variations such as permutating the entries of mtable 
(and mt) will give the same mapping. So will other ways of 
representing the array of offsets, e.g., as two 32-bit 
words instead of four 16-bit words. 

/* maptable */ 

#define MAPVECLEN 4 

typedef uintlS MAPVEC[MAPVECLEN] ; 

typedef MAPVEC MCOMPACT; 

MCOMPACT mt[TMAX]; 

/* mt is the maptable used during lookup. 

initialized from mtable which is used while building. 



typedef struct mentry { 
uintl6 mask; 



/* 16 bit pattern (LSB is always 

set!) */ 
/* 8 bit len (value 1-16) */ 



uintl6 len; 
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MAPVEC map; /* 16 4-bit offsets in groups of 

four */ 

} MENTRY, *MP; 

5 void mentry2mcompact (MENTRY *from, MCOMPACT +to) 

/* takes the map part from 'from' and puts it in 'to' */ 

{ 

int i; 

10 for(i=0; KMAPVECLEN; i++) { 

(*to) [i] = f rom->map[i]; 

} 

} 

15 extern void mtable_compact { ) 

/* initializes mt from mtable */ 
{ 

register int i; 

20 for{i=0; i<TMAX; i++) { 

mentry2mcompact (&mtable[i], &mt[i]) ; 

} 

} 

25 Levels two and three of the data structure consist of 

chunks. A chunk covers a subtree of height 8 and can 
contain at most 2 s = 256 heads. A root head in level n-1 
points to a chunk in level n. 

There are three varieties of chunks depending on how 

30 many heads the imaginary bit-vector contains. When there 
are 

1-8 heads, the chunk is sparse and is represented by 
an array of the 8-bit indices of the heads, plus eight 16- 
bit pointers; a total of 24 bytes. 
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9-64 heads, the chunk is dense. It is represented 
analogously with level 1, except for the number of base 
indexes. The difference is that only one base index is 
needed for all 16 code words, because 6-bit offsets can 
5 cover all 64 pointers. A total of 34 bytes are needed, plus 
IB to 128 bytes for pointers. 

65-256 heads, the chunk is very dense. It is 
represented analogously with level 1. 16 code words and 4 
base indexes give a total of 40 bytes. In addition the 65 
10 to 256 pointers require 130 to 512 bytes. 

Dense and very dense chunks are searched analogously 
with the first level. 

Sparse chunks are searched with a special-purpose 
binary search tailormade for 8 elements. This is 
15 considerably faster than a linear search and a general- 
purpose binary search. Moreover, this search can be 
implemented without using conditional jumps on processor 
architectures that have a conditional move instruction. 

20 /* search function for sparse chunks */ 

static inline uintl6 f indsparse ( SPARSECHUNK *chu, uint32 

val) 

25 /* chu->vals is a sorted array 0..7 of 8-bit values 

chu->rinfo is the corresponding array 0..7 of pointers 
val is the search key 

*/ 
30 { 

uint8 *p, *q; 

p = q = & (chu->vals[0]) ; 



35 



P += (Mp + 3) > val) « 2; 
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P + 



P + 



{*(p + 1) > val) « 1; 
<+p > val); 



return (chu->rinf o[p - q]) ; 



10 



15 



20 



25 



30 



Dense and very dense chunks are optimized analogously 
with level 1, as described. In sparse chunks, two 
consecutive heads can be merged and represented by the 
smaller if their next-hops are identical. When deciding 
whether a chunk is sparse or dense, this merging is taken 
into account so that the chunk is deemed sparse when the 
number of merged heads is 8 or less. Many of the leaves 
that were added to make the tree full will occur in order 
and have identical next-hops. Heads corresponding to such 
leaves will be merged in sparse chunks. 

This optimization shifts the chunk distribution from 
the larger dense chunks towards the smaller sparse chunks. 
For large tables, the size of the forwarding table is 
typically decreased by 5 to 15 per cent. 

The data structure can accommodate considerable 
growth in the number of routing entries. There are two 
limits in the current design. 

1. The number of chunks of each kind is limited to 
2 14 16384 per level. 

Table 1 shows that this is about 16 times more than 
is currently used. If the limit is ever exceeded, the data 
structure can be modified so that pointers are encoded 
differently to give more room for indices, or so that the 
pointer size is increased. 

2. The number of pointers in levels two and three is 
limited by the size of the base indices. 

The current implementation uses 16-bit base indices 
and can accommodate a growth factor of 3 to 5. If the limit 
is exceeded it is straightforward to increase the size of 
base pointers to three bytes. The chunk size is then 
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increased by 3 per cent for dense chunks and 10 per cent 
for very dense chunks. Sparse chunks are not affected. 

It is clear that the data structure can accommodate a 
large increase in number of routing entries. Its size will 

5 grow almost linearly with the number of routing entries. 

To investigate the performance of the forwarding 
tables, a number of IP routing tables were collected. 
Internet routing tables are currently available at the web 
site for the Internet Performance Measurement and Analysis 

10 (IPMA) project (http://www.ra.net/statistics/), and were 
previously made available by the now terminated Routing 
Arbiter project (http://www.ra.net/statistics/). The 
collected routing tables are daily snapshots of the routing 
tables used at various large Internet interconnection 

15 points. Some of the routing entries in these tables contain 
multiple next-hops. In that case, one of them was randomly 
selected as the next-hop to use in the forwarding table. 

Table 1 in FIG 8 shows data on forwarding tables 
constructed from various routing tables. For each site, it 

20 shows data and results for the routing table that generated 
the largest forwarding table. Routing entries is the number 
of routing entries in the routing table, and next-hops is 
the number of distinct next-hops found in the table. Leaves 
is the number of leaves in the prefix tree after leaves 

25 have been added to make it full. 

Build time in Table 1 is the time required to 
generate the forwarding table from an in-memory binary tree 
representation of the routing table. Times were measured on 
a 333 MHz Alpha 21164 running DEC OSF1. Subsequent columns 

30 show the total number of sparse, dense, and very dense 
chunks in the generated table followed by the number of 
chunks in the lowest level of the data structure. 

It is clear from Table 1 that new forwarding tables 
can be generated quickly. At a regeneration frequency of 

35 one Hz, less than one tenth of the Alpha's capacity is 
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consumed. As described above, higher regeneration frequen- 
cies than 1 Hz are not required. 

The larger tables in Table 1 do not fit entirely in 
the 96 Kbytes secondary cache of the Alpha. It is feasible, 

5 however, to have a small amount of very fast SRAM in the 
third level cache for the pieces that do not fit in the 
secondary cache, and thus reduce the cost of a miss in the 
secondary cache. With locality in traffic patterns, most 
memory references would be to the secondary cache. 

10 An interesting observation is that the size of these 

tables are comparable to what it would take to just store 
all prefixes in an array. For the larger tables, no more 
than 5.6 bytes per prefix are needed. More than half of 
these bytes are consumed by pointers. In the Sprint table 

15 there are 33469 pointers that require over 65 Kbytes of 
storage. It is clear that further reductions of the 
forwarding table size could be accomplished by reducing the 
number of pointers. 

Measurements on the lookup routine are cone on a C 

20 function compiled with the GNU C-compiler gcc (disclosed by 
the Using and porting gnu cc. Manual, Free Software 
Foundation, November 1995, ISBN 1-882114-66-3) . Reported 
times do not include the function call or the memory access 
to the next-hop table, gcc generates a code that uses 

25 approximately 50 Alpha instructions to search one level of 
the data structure in the worst case. On a Pentium Pro, gcc 
generates a code that uses 35 to 4 5 instructions per level 
in the worst case. 

The following C-code function shows the code of the 

30 lookup function used in the measurements. 

The level 1 codeword array is called intl and the 
base index array basel. The pointers in level 1 are stored 
in the array htabl. 

Chunks are stored in the arrays cis, cid, cidd, where 

35 i is the level. 
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Base addresses and pointers for dense and very dense 
chunks are stored in the arrays baseid, baseidd, and 
htabid, htabidd, respectively. 

5 /* lookup function for forwarding table */ 

#include "conf.h" 
#include "forward. h M 
tinclude "mtentry.h" 
10 #include "mtable.h" 

#include "bit2index.h" 

#define TABLE_LOOKUP 

15 #include "sparse. h" 
#include "timing. h" 
#include "lookup. h" 

/* these macros work only if ip is a 32-bit unsigned int 
20 (uint32) */ 

#define EXTRACT {start, bits, ip) ( ( (ip)«(start) )»(32- 
(bits) ) ) 

tdefine GETTEN (m) ( { (m) «22 ) »22 ) 
25 tdefine GETSIX(m) {(m)»10) 

#define bit2o(ix, bit) 

{ (mt[ (ix) ] [ <bit}»2]»( ( (bit) &0x3)«2) ) &0xf ) 

30 /* lookup (ipaddr) — index to routing table entry for 
ipaddr */ 

unsigned int lookup (uint32 ipaddr) 
{ 

35 
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uint32 


ix; 


/* 


index to codeword array 


*/ 


uint32 


code ; 


/* 


16 bit codeword 


*/ 


int32 


diff ; 


/* 


difference from TMAX 


*/ 


uint32 


ten; 


/* 


index to mt 


*/ 


uint32 


six; 


/* 


six 'extra' bits of code 


*/ 


int32 


nhop; 


/* 


pointer to next-hop 


*/ 


uint32 


off ; 


/* 


pointer offset 


*/ 


uint32 


hbase; 


/* 


base for hash table 


V 


uint32 


pntr; 


/* 


hash table entry 


*/ 


int32 


kind; 


/* 


kind of pntr 


*/ 


uint32 


chunk; 


/* 


chunk index 


*/ 


uint8 


*p,*q; 


/* 


pointer to search sparse 


*/ 


uint32 


key; 


/* 


search key in sparse 


*/ 



15 /* timed from *HERE* */ 

ix = EXTRACT (0, 12, ipaddr) ; 
code = intl [ix] ; 
ten = GETTEN (code) ; 
20 six = GETS IX (code) ; 

if ( (diff={ten-TMAX) )>=0) { 

nhop = (six I (dif f«6) ) ; 
25 goto end; 

} 

off = bit2o(ten, EXTRACT (12,4, ipaddr ) ) ; 

hbase - basel[ix»2]; 

pntr = htabl [hbase+six+of f ] ; 

30 

if ((kind = (pntr & 0x3))) { 
chunk = (pntr»2) ; 



35 



if (--kind) { 
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if ( — kind) { 



15 



key = EXTRACT (16, 8, ipaddr) ; 
p = q = c2s [chunk] .vals [0] ) ; 

if ( *(p+3) > key) ( 
p += 4; 

}; 



10 while ( *p > key) { 

P++; 

}; 



pntr - c2s [chunk] . rinfo [p - q] ; 
} else { 



ix = EXTRACT (16, 4, ipaddr) ; 
code = c2dd [chunk] [ix] ; 
20 ten = GETTEN (code) ; 

six « GETSIX(code) ; 



if ( (diff=(ten-TMAX) )>=0) { 

25 nhop = (six | (diff«6)); 

goto end; 

} 

hbase = htab2ddbase [ (chunk«2) j (ix»2) ]; 
off = bit2o(ten, EXTRACT (20 , 4 , ipaddr )) ; 
30 pntr - htab2dd[hbase+six+of f ] ; 

} 

} else { 



35 



ix = EXTRACT (16,4, ipaddr ) ; 
code = c2d [chunk] [ix] ; 
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ten ^ GETTEN (code) ; 
six = GETSIX (code) ; 

if ( (dif f=(ten-TMAX) ) >=0) { 

5 

nhop = (six | (dif f«6) ) ; 
goto end; 

} 

hbase = htab2dbase [chunk] ; 
10 off = bit2o(ten, EXTRACT { 16, 4 , ipaddr) ) ; 

pntr = htab2d[hbase+six+of f ] ; 



15 



20 



if {(kind = (pntr & 0x3})) { 

chunk = (pntr»2) ; 

if ( — kind) { 
If ( — kind) { 

key = EXTRACT (24,8, ipaddr ) ; 
p = q = c3s [chunk] .vals [0] ) ; 



if { Mp+3) > key) { 
25 p 4; 

}; 

while ( *p > key) { 
P++; 

30 }; 

pntr = c3s [chunk] . rinfo [p - q] ; 



35 



} else { 
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ix = EXTRACT (24, 4, ipaddr) ; 
code = c3dd[chunk) [ix) ; 
ten = GETTEN (code) ; 
six = GETSIX(code) ; 

if ( (diff=(ten-TMAX) )>=0) { 

nhop = (six | (dif f <<6) ) ; 
goto end; 

} 

off = bit2o(ten, EXTRACT (28,4, ipaddr) ) ; 
hbase = htab3ddbase[ (chunk«2) I (ix»2) ] ; 
pntr = htab3dd[hbase+six+of f ] ; 

1 

} else { 

ix = EXTRACT (24,4, ipaddr) ; 
code = c3d [chunk] [ix] ; 
ten = GETTEN (code) ; 
20 six = GETSIX(code) ; 

if ( (diff={ten-TMAX) )>=0) { 

nhop = (six I (diff«6)); 
goto end; 

} 

off = bit2o(ten, EXTRACT (28,4, ipaddr ) ) ; 
hbase = htab3dbase [chunk) ; 
pntr = htab3d[hbase+six+of f ] ; 

} 

} 

} 

nhop = (pntr >> 2) ; 
35 end: 



10 



15 



25 



30 
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/+ timed to HERE! */ 



return nhop; 



10 



15 



20 



25 
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It is possible to read the current value of the clock 
cycle counter on Alphas and Pentium Pros. This facility has 
been used to measure lookup times with high precision: one 
clock tick is 5 nanoseconds at 200 MHz and 3 nanoseconds at 
333 Mhz. 

Ideally, the entire forwarding table would be placed 
in cache so lookups would be performed with an undisturbed 
cache. That would emulate the cache behaviour of a 
dedicated forwarding engine. However, measurements have 
been performed on conventional general-purpose workstations 
only and it is difficult to control the cache contents on 
such systems. The cache is disturbed whenever I/O is 
performed, an interrupt occurs, or another process starts 
running. It is not even possible to print out measurement 
data or read a new IP address from a file without 
disturbing the cache. 

The method used performs each lookup twice and the 
lookup time for the second lookup is measured. In this way, 
the first lookup is done with a disturbed cache and the 
second in a cache where all necessary data have been forced 
into the primary cache by the first lookup. After each pair 
of lookups measurement data are printed out and a new 
address is fetched, a procedure that again disturbs the 
cache . 

The second- lookup will perform better than lookups in 
a forwarding engine because data and instructions have 
moved into the primary cache closest to the processor. To 
get an upper limit on the lookup time, the additional time 
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required for memory accesses to the secondary cache must be 
added to the measured times. To test all paths through the 
forwarding table, lookup time was measured for each entry 
in the routing table, including the entries added by the 

5 expansion to a full tree. 

Average lookup times can not be inferred from these 
experiments because it is not likely that a- realistic 
traffic mix would have a uniform probability for accessing 
each routing entry. Moreover, locality in traffic patterns 

10 will keep frequently accessed parts of the data structure 
in the primary cache and, thus, reduce the average lookup 
time. The performance figures calculated below are conser- 
vative because it is assumed that all memory accesses miss 
in the primary cache, and that the worst case execution 

15 time will always occur. Realistic lookup speeds would be 
higher . 

Table 1 shows that there are very few chunks in level 
three of the data structure. That makes it likely that the 
vast majority of lookups need to search no more than two 

20 levels to find the next hop. Therefore, the additional time 
for memory accesses to the secondary cache is calculated 
for eight instead of the worst-case twelve memory accesses. 
If a significant fraction of lookups were to access those 
few chunks, they would migrate into the primary cache and 

25 all twelve memory accesses would become less expensive. 

An experiment was performed on an Alpha 21164 with a 
clock frequency of 333 MHz; one cycle takes 3 nanoseconds. 
Accesses to the 8 Kbytes primary data cache completes in 2 
cycles and accesses to the secondary 96 Kbytes cache 

30 require 8 cycles. See Table 2 in FIG 9. 

FIG 10 shows the distribution of clock ticks elapsed 
during the second lookup for the Alpha on the Sprint 
routing table from January 1st. The fastest observed 
lookups require 17 clock cycles. This is the case when the 

35 code word in the first level directly encodes the index to 
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the next-hop table. There are very few such routing 
entries. However, as each such routing entry covers many IP 
addresses, actual traffic may contain many such destination 
addresses. Some lookups take 22 cycles, which must be the 

5 same case as the previous are. Experiments have confirmed 
that when the clock cycle counter is read with two con- 
secutive instructions, the difference is sometimes 5 cycles 
instead of the expected 0. 

The next spike in FIG 10 is at 41 clock cycles, which 

10 is the case when the pointer found in the first level is an 
index to the next-hop table. Traditional class B addresses 
fall in this category. Spikes at 52 — 53, 57, 62, 67, and 72 
ticks correspond to finding the pointer after examining 
one, two, three, four, or five values in a sparse level 2 

15 chunk. The huge spikes at 75 and 83 ticks are because that 
many ticks are required to search a dense and very dense 
chunk, respectively. A few observations above 83 correspond 
to pointers found after searching a sparse level 3 chunk, 
probably due to variations in execution time. Cache 

20 conflicts in the secondary cache, or differences in the 

state of pipelines and cache system before the lookup, can 
cause such variations. The tail of observations above 100 
clock cycles is either due to such variations or to cache 
misses. 300 nanoseconds should be sufficient for a lookup 

25 when all data are in the primary cache. 

The difference between a data access in the primary 
cache and the secondary cache is 8-2 = 6 cycles. Thus, 
searching two levels of the data structure in the worst 
case requires 8x6 = 48 clock cycles more than indicated by 

30 FIG 10. That means at most 100+48 = 148 cycles or 444 
nanoseconds for the worst case lookup when 2 levels are 
sufficient. Thus the Alpha should be able to do at least 
2.2 million routing lookups per second with the forwarding 
table in the secondary cache. 
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Another experiment was performed on a Pentium Pro 
with a clock frequency of 200 MHz ; one cycle takes 5 
nanoseconds. The primary 8 Kbytes cache has a latency of 2 
cycles and the secondary cache of 256 Kbytes has a latency 

5 of 6 cycles. See Table 2. 

FIG 11 shows the distribution of clock ticks elapsed 
during the second lookup for the Pentium Pro with the same 
forwarding table as for the Alpha 21164. The sequence of 
instructions that fetches the clock cycle counter takes 33 

10 clock cycles. When two fetches occur immediately after each 
other the counter values differ by 33. For this reason, all 
reported times have been reduced by 33. 

The fastest observed lookups are 11 clock cycles, 
about the same speed as for the Alpha 21164. The spike 

15 corresponding to the case when the next-hop index is found 
immediately after the first level occurs at 25 clock 
cycles. The spikes corresponding to a sparse level 2 chunk 
are grouped closely together in the range 36 to 40 clock 
cycles. The different caching structure of the Pentium 

20 seems to deal better with linear scans than the caching 
structure of the Alpha 21164. 

When the second level chunks are dense and very 
dense, the lookup requires 48 and 50 cycles, respectively. 
There are some additional irregular spikes up to 69, above 

25 which there are very few observations. It is clear that 69 
cycles (345 nanoseconds) is sufficient to do a lookup when 
all data are in the primary cache. 

The difference in access time between the primary and 
secondary cache is 20 nanoseconds {4 cycles) . The lookup 

30 time for the Pentium Pro when two levels need to be 

examined is then at worst 69 + 8x4 = 101 cycles or 505 
nanoseconds. The Pentium Pro can do at least 2.0 million 
routing lookups per second with the forwarding table in 
secondary cache. 
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It should be apparent that the present invention 
provides an improved IP routing lookup method and system 
that fully satisfies the aims and advantages set forth 
above. Although the invention has been described in 

5 conjunction with specific embodiments thereof, alterna- 
tives, modifications and variations are apparent to those 
skilled in the art. 

In the above described embodiment the lookup function 
is implemented in C programming language. For those skilled 

10 in the art of programming it will be clear that other pro- 
gramming languages can be used. Alternatively, the lookup 
function can be implemented in hardware using standard 
digital design techniques, as also will be clear to those 
skilled in the art of hardware design. 

15 For example, the invention is applicable for other 

computer system configurations than systems using the Alpha 
21164 or Pentium Pro, other ways to cut the prefix tree, 
different number of levels in the tree, other ways to 
represent the maptable, and other ways to encode the code 

20 words . 

Additionally, the invention can by utilized for Fire- 
wall routing lookups . 
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CLAIMS 
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15 



20 
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1. A method of IP routing lookup in a routing table, 
comprising entries of arbitrary length prefixes with 
associated next-hop information in a next-hop table, to 
determine where IP datagrams are to be forwarded, 
characterized by the steps of 

storing in storage means a representation of the 
routing table, in the form of a complete prefix tree {7), 
defined by the prefixes of all routing table entries, 
completed such that each node has either no or two 
children, all children added are leaves with the same next- 
hop information as the closest ancestor with next -hop 
information, or an undefined next -hop if no such ancestor 
exists , 

storing in said storage means a representation of a 
bit vector (8) , comprising data of a cut through the prefix 
tree (7) at a current depth (D) with one bit per possible 
node at the current depth, wherein the bit is set when 
there is a node in the prefix tree (7) , 

storing in said storage means an array of pointers, 
£or genuine heads an index to the next -hop table, and for 
root heads an index to a next -level chunk, 

dividing said bit -vector (8) into bit -masks of a 
certain length, 

storing in said storage means a representation of the 
possible bit-masks in a maptable, 

storing in said storage means an array of code words, 
each encoding a row index into the maptable and a pointer 
offset , 

storing in said storage means an array of base 
addresses , 

accessing a code word at a location corresponding to 
a first index part (ix) of the IP address in the array of 
code words, 
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accessing a maptable entry part at a location 
corresponding to a column index part (bit) of the IP 
address and a row index part (ten) of the code word in the 
maptable, 

5 accessing a base address at a location corresponding 

to the second index part (bix) of the IP address in the 

array of base addresses, and 

accessing a pointer at a location corresponding to 

the base address plus a pointer offset (six) of the 
10 codeword plus the maptable entry part in the array of 

pointers . 

2. A system for IP routing lookup in a routing table, 
comprising entries of arbitrary length prefixes with 
associated next-hop information in a next-hop table, to 

15 determine where IP datagrams are to be forwarded, 

characterized by a routing table, in the form 
of a complete prefix tree (7) , defined by the prefixes of 
all routing table entries, each node has either no or two 
children, all children added are leaves with the same next- 

20 hop information as the closest ancestor with next -hop 

information, or an undefined next -hop if no such ancestor 
exists, 

a representation of a bit vector (8) , comprising data 
of a cut through the prefix tree (7) at a current depth (D) 
25 with one bit per possible node at the current depth, 

wherein the bit is set when there is a node in the prefix 
tree (7), 

an array of pointers, for genuine heads an index to 
the next -hop table, and for root heads an index to a next- 
30 level chunk, 

said bit-vector (8) divided into bit-masks of a 
certain length, 

a maptable comprising a representation of the 
possible bit -masks, 
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an array of code words, each encoding a row index 
into the maptable and a pointer offset, and 
an array of base addresses . 
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